组合模式——七种结构型模式之一

1.前言


组合是一种整体与部分的关系,即对象与其内部对象之间的关系。通过之前的外观模式,体会到对象内部是可以很复杂的。最常见的情况便是,一个对象由多个对象组合而成,而部分参与组合的对象又是由多个对象组合而成,很像是树状结构。

这种结构的对象该如何访问呢?作为开发人员肯定想忽略掉这些区别,以一种统一的方式来处理每个对象。

2.概念


组合模式将对象组合成树形结构以表示“整体-部分”的层次结构,使得用户对组合对象和单个对象的使用具有一致性。这种统一的风格便是抽象的体现,需通过抽象类或接口来定义,由各种对象去实现。

这种模式特别适合模块化的结构,即层级结构明显,功能划分细致,像公司的组织架构,又或者计算机的文件系统。

3.场景


一个大的集团公司通常都是由许多子公司组成,每个子公司都有自己的业务范围。但只要是公司,必然有不同的部门,每个部门都有自己的工作目标。只不过,总公司统筹所有子公司而已。

4.写法


// 1.定义访问所有对象的共有接口
public abstract class Function {

    protected String name;

    // 2.实现缺省行为
    public Function(String name) {
        this.name = name;
    }

    public abstract void work();
}
// 1.实现组合对象及共有行为
public class Company extends Function {

    private List<Function> mFunctions = new ArrayList<>();

    public Company(String name) {
        super(name);
    }

    @Override
    public void work() {
        System.out.println("公司名为 " + name);
        if (mFunctions != null && !mFunctions.isEmpty()) {
            for (Function function : mFunctions) {
                function.work();
            }
        }
    }

    // 2.组合对象的特有行为
    public void addFunction(Function function) {
        mFunctions.add(function);
    }

    public void removeFunction(Function function) {
        mFunctions.remove(function);
    }

    public Function getFunction(int index) {
        return mFunctions.get(index);
    }
}
// 1.实现单个对象及共有行为
public class Department extends Function {
    public Department(String name) {
        super(name);
    }

    @Override
    public void work() {
        System.out.println("部门名为 " + name);
    }

    // 2.可添加单个对象的特有行为
}
public class Client {
    public static void main(String[] args) {
        // 1.构造集团公司
        Company group = new Company("XXX集团");

        Company company = new Company("XXX公司");
        Department onlineShopping = new Department("网络购物事业部");
        Department mobile = new Department("移动事业部");

        group.addFunction(company);
        group.addFunction(onlineShopping);
        group.addFunction(mobile);

        // 2.构造子公司
        Company subcompany = new Company("XXX子公司");
        Department localLife = new Department("本地生活事业部");
        Department cloud = new Department("云产品事业部");

        company.addFunction(subcompany);
        company.addFunction(localLife);
        company.addFunction(cloud);

        group.work();
    }
}

通过上面的代码,细心的朋友肯定会发现一个问题,那就是违背了依赖倒置原则。在Client类中直接使用Function的具体实现类,增加了代码间的耦合度,还忽视了Function的抽象作用。目前主流的开发方式就是面向接口编程,除了把焦点放在接口的设计外,还应注重接口的使用。

Function类无法使用,关键是因为只声明了公司和部门的共有方法,在实现类中才添加特有方法,对调用者屏蔽了具体实现的细节,这是安全的组合模式。若希望Function类得到使用,可以让其包括所有特有方法的声明,对外公开完整的访问结构,称为透明的组合模式

public abstract class Function {

    protected String name;

    public Function(String name) {
        this.name = name;
    }

    public abstract void work();

    // 添加Company特有方法的声明
    public abstract void addFunction(Function function);

    public abstract void removeFunction(Function function);

    public abstract Function getFunction(int index);
}
public class Department extends Function {
    public Department(String name) {
        super(name);
    }

    @Override
    public void work() {
        System.out.println("部门名为 " + name);
    }

    // 1.实现Company特有方法
    @Override
    public void addFunction(Function function) {
        throw new UnsupportedOperationException("No function");
    }

    @Override
    public void removeFunction(Function function) {
        throw new UnsupportedOperationException("No function");
    }

    @Override
    public Function getFunction(int index) {
        throw new UnsupportedOperationException("No function");
    }

    // 2.可添加单个对象的特有行为
}

5.总结


在实际开发中,尤其是安卓开发,SDK提供的UI组件架构设计就是使用组合模式。大家可以回顾一下,TextView和ViewGroup继承自View,而共有方法包括了onMeasure()onLayout()onDraw()等。之所以这样,除了看中组合模式便于访问结构树中的节点外,还利于增删节点,为树形结构的面向对象实现提供灵活的解决方案。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,569评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,499评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,271评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,087评论 0 209
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,474评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,670评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,911评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,636评论 0 202
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,397评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,607评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,093评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,418评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,074评论 3 237
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,092评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,865评论 0 196
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,726评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,627评论 2 270

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 170,569评论 25 707
  • 工厂模式类似于现实生活中的工厂可以产生大量相似的商品,去做同样的事情,实现同样的效果;这时候需要使用工厂模式。简单...
    舟渔行舟阅读 7,622评论 2 17
  • 1 场景问题# 1.1 商品类别树## 考虑这样一个实际的应用:管理商品类别树。 在实现跟商品有关的应用系统的时候...
    七寸知架构阅读 5,828评论 10 59
  • 今天又再次整理了餐桌。 不知道是不是因为餐桌是全家人在家中每天最常光顾的一个地方,几天时间就又乱了,没关系,乱了我...
    风吹才会铃动吗阅读 91评论 0 0
  • 那是一丛从坚硬岩缝里杀出来的生命,单枪匹马地与大自然挑战。 曾几何时,在一本世界新闻杂志上看到一幅图片,在一双焦黑...
    艺燃阅读 198评论 0 2